package com.test.cordova.util;

import android.Manifest;
import android.app.Activity;
import android.content.Context;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Environment;
import android.os.Looper;
import android.support.v4.app.ActivityCompat;
import android.util.Log;
import android.widget.Toast;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.Writer;
import java.lang.reflect.Field;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * 日志存储本地
 */

public class CrashHandler implements Thread.UncaughtExceptionHandler {

    // 用于打印日志的 TAG 标识符
    public static final String TAG = "CrashHandler";
    private static final int WRITE_EXTERNAL_CODE = 145;
    // 系统默认的UncaughtException处理类
    private Thread.UncaughtExceptionHandler mDefaultHandler;
    // 程序的Context对象
    private Activity mActivity;
    // 用来存储设备信息和异常信息
    private Map<String, String> mInfos = new HashMap<String, String>();
    // 用于格式化日期,作为日志文件名的一部分
    private DateFormat formatter = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss");

    // 单例模式
    private static CrashHandler INSTANCE = new CrashHandler();
    private CrashHandler() {
    }
    public static CrashHandler getInstance() {
        return INSTANCE;
    }

    /**
     * 初始化该类, 向系统中注册
     * @param activity
     */
    public void init(Activity activity) {
        mActivity = activity;
        // 获取系统默认的 UncaughtException 处理器
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
        // 设置该 CrashHandler 为程序的默认处理器
        Thread.setDefaultUncaughtExceptionHandler(this);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (ActivityCompat.checkSelfPermission(mActivity, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
                ActivityCompat.requestPermissions(mActivity, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, WRITE_EXTERNAL_CODE);
            }
        }
    }

    /*
     * 出现未捕获的异常时, 会自动回调该方法
     * (non-Javadoc)
     * @see java.lang.Thread.UncaughtExceptionHandler#uncaughtException(java.lang.Thread, java.lang.Throwable)
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
		/*
		 * 调用 handleException() 方法处理该线程
		 * 如果返回 true 说明处理成功, 如果返回 false 则调用默认的异常处理器来处理
		 * 一般情况下该方法都会成功处理
		 */
        if (!handleException(ex) && mDefaultHandler != null) {
            // 如果用户没有处理则让系统默认的异常处理器来处理
            mDefaultHandler.uncaughtException(thread, ex);
            Logger.e(TAG, "error : "+ex.getMessage());
        } else {
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                Logger.e(TAG, "error : "+e.getMessage());
            }
            // 退出程序
            android.os.Process.killProcess(android.os.Process.myPid());
            System.exit(1);
        }
    }

    private Throwable throwable = null;
    /**
     * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成.
     * @param ex
     * 		异常信息
     * @return
     * 		true:如果处理了该异常信息;否则返回false.
     */
    private boolean handleException(Throwable ex) {
        if (ex == null) {
            return false;
        }
		/*
		 * 使用Toast来显示异常信息,
		 * 由于在主线程会阻塞,
		 * 不能实时出现 Toast 信息,
		 * 这里我们在子线程中处理 Toast 信息
		 */
        /*new Thread() {
            @Override
            public void run() {
                Looper.prepare();
                Toast.makeText(mActivity, "很抱歉,程序出现异常,即将退出！", Toast.LENGTH_LONG).show();
                Looper.loop();
            }
        }.start();*/
        throwable = ex;
        // 收集设备参数信息
        collectDeviceInfo(mActivity);
        // 保存日志文件
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (ActivityCompat.checkSelfPermission(mActivity, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
                ActivityCompat.requestPermissions(mActivity, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, WRITE_EXTERNAL_CODE);
            } else {
                saveCrashInfoFile(ex);
            }
        } else {
            // 保存日志文件
            saveCrashInfoFile(ex);
        }

        return true;
    }

    /**
     * 收集设备参数信息, 将手机到的信息存储到
     * @param ctx
     * 		上下文对象
     */
    private void collectDeviceInfo(Context ctx) {
        try {
            //获取包管理器
            PackageManager pm = ctx.getPackageManager();
            //获取包信息
            PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(), PackageManager.GET_ACTIVITIES);
            if (pi != null) {
                //版本号
                String versionName = pi.versionName == null ? "null" : pi.versionName;
                //版本代码
                String versionCode = pi.versionCode + "";
                //将版本信息存放到 成员变量 Map<String, String> mInfos 中
                this.mInfos.put("versionName", versionName);
                this.mInfos.put("versionCode", versionCode);
            }
        } catch (PackageManager.NameNotFoundException e) {
            Logger.e(TAG, "an error occured when collect package info :"+e.getMessage());
        }

        //获取 Build 中定义的变量, 使用反射方式获取, 该类中定义了设备相关的变量信息
        Field[] fields = Build.class.getDeclaredFields();
        //遍历获取额变量, 将这些信息存放到成员变量 Map<String, String> mInfos 中
        for (Field field : fields) {
            try {
                //设置 Build 成员变量可访问
                field.setAccessible(true);
                //将 设备相关的信息存放到 mInfos 成员变量中
                mInfos.put(field.getName(), field.get(null).toString());
                Logger.d(TAG, field.getName() + " : " + field.get(null));
            } catch (Exception e) {
                Logger.e(TAG, "an error occured when collect crash info:"+ e.getMessage());
            }
        }
    }

    /**
     * 保存错误信息到文件中
     * @param ex
     * @return 返回文件名称,便于将文件传送到服务器
     */
    private String saveCrashInfoFile(Throwable ex) {
        //存储相关的字符串信息
        StringBuffer sb = new StringBuffer();
        //将成员变量 Map<String, String> mInfos  中的数据 存储到 StringBuffer sb 中
        for (Map.Entry<String, String> entry : this.mInfos.entrySet()) {
            String key = entry.getKey();
            String value = entry.getValue();
            sb.append(key + "=" + value + "\n");
        }

        //将 StringBuffer sb 中的字符串写出到文件中
        Writer writer = new StringWriter();
        PrintWriter printWriter = new PrintWriter(writer);
        ex.printStackTrace(printWriter);
        Throwable cause = ex.getCause();
        while (cause != null) {
            cause.printStackTrace(printWriter);
            cause = cause.getCause();
        }
        printWriter.close();
        String result = writer.toString();
        sb.append(result);
        try {
            long timestamp = System.currentTimeMillis();
            String time = formatter.format(new Date());
            String fileName = "crash-" + time + "-" + timestamp + ".txt";

            if (Environment.getExternalStorageState().equals(
                    Environment.MEDIA_MOUNTED)) {
                //获取文件输出路径
                saveExternalStorageCache(sb, fileName);
            }
            return fileName;
        } catch (Exception e) {
            Logger.e(TAG, "an error occured while writing file...: "+e.getMessage());
        }
        return null;
    }


    /**
     * 外部存储路径
     * @param sb
     * @param fileName
     * @throws IOException
     */
    private void saveExternalStorageCache(StringBuffer sb, String fileName) throws IOException {
        //异常日志输出文件路径： storage/emulated/0/Android/data/包名
        String path = StorageUtil.getCacheDirectory(mActivity).getPath();
        //创建输出流
        FileOutputStream fos = new FileOutputStream(path + fileName);
        //向文件中写出数据
        fos.write(sb.toString().getBytes());
        fos.close();

    }

    /**
     * 处理请求权限
     * @param requestCode
     * @param permissions
     * @param grantResults
     */
    public void handleRequestPermission(int requestCode, String[] permissions, int[] grantResults){
        Logger.i(TAG,"handleRequestPermission");
        if (requestCode == WRITE_EXTERNAL_CODE) {
            if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // 保存日志文件
                if (throwable != null) {
                    saveCrashInfoFile(throwable);
                }
            } else  {
                  if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
                        //上次弹出权限点击了禁止（但没有勾选“下次不在询问”） true
                        ActivityCompat.requestPermissions(mActivity, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, WRITE_EXTERNAL_CODE);
                    } else {
                        // 上次选择禁止并勾选：下次不在询问	false
                        Toast.makeText(mActivity,"请打开存储权限！",Toast.LENGTH_SHORT).show();
                    }
                Logger.e(TAG,"handleRequestPermission: denied caches");
            }
        }
    }

}
